Android 如何通过google play获取最新版本并更新当前版本

16 篇文章 0 订阅
2 篇文章 0 订阅

前言

当遇到直接通过Google play上的版本名称来更新我们自己的App时,这时我们可以通过访问Google play上目标App的版本信息,通知自己的App,并跳转到Google play。下面来看具体实现。

一、工具集成

implementation 'org.jsoup:jsoup:1.10.2'

ps:jsoup官网github源码文档这个工具主要是java代码,作为抓取HTML的工具,有很高的效率,如果有兴趣可以通过文档详细了解这个工具。我们可以通过他获得网页上想要的信息,正如现在我们需要“current version”信息。

二、代码逻辑

            val document: Document = Jsoup.connect(
                "https://play.google.com/store/apps/details?id=" + (if (!BuildConfig.DEBUG) BuildConfig.APPLICATION_ID else testPackageName) + "&hl=en"
            ).timeout(30000).userAgent("Mozilla/5.0 (Windows; U; WindowsNT 5.1; en-US; rv1.8.1.6) Gecko/20070725 Firefox/2.0.0.6").referrer("http://www.google.com").get()

其中connect里面为要访问的目标url,拼接的id为访问目标的app id,通常为包名。这里可以直接获取BuildConfig.APPLICATION_ID当前项目id,如果在测试的时候没有上架到google play,可以找个已上架的app,作为测试id。
timeout为链接超时时间,userAgent为当前链接需要设置的代理信息。

三、获取目标信息

  val element: Elements = document.getElementsContainingOwnText("Current Version")
            for (ele in element) {
                Logg.d(tag, "ele:$ele")
                if (ele.siblingElements() != null) {
                    Logg.d(tag, "ele.siblingElements:${ele.siblingElements()}")
                    val sibElemets: Elements = ele.siblingElements()
                    for (sibElemet in sibElemets) {
                        newVersion = sibElemet.text()
                    }
                }
            }

通过第二步的操作,成功了的话我们可以获得所有的HTML信息并放入document中。此时只需要输入我们需要元素key值,再轮巡取出目标text,这就是本次所要的version值。当然,你也可以根据需要分析document,在这些信息取出需要的信息。

四、跳转Google play

       activity?.let { it ->
            val intentData = Intent(Intent.ACTION_VIEW).setData(
                Uri.parse(
                    "market://details?id=" + if (!BuildConfig.DEBUG) it.packageName else testPackageName
                )
            )
            if (it.packageManager?.let { intentData.resolveActivity(it) } != null) {
                it.startActivity(intentData)
            } else {
                intentData.data = Uri.parse(
                    "https://play.google.com/store/apps/details?id=" + if (!BuildConfig.DEBUG) it.packageName else testPackageName
                )
                it.startActivity(intentData)
            }
        }

如果顺利的话,第三步我们已经拿到了关键的version name,此时只要根据自己的逻辑判断,跳转到Google play就可以了。当然,如果当前设备没有安装Google play,那就跳转浏览器。这里的id便是跳转目标app的包名packageName.

五、代码全览

工具类

object VersionUpdateUtil {
    private val tag = "${this.javaClass.simpleName}=="
    private var newVersion: String? = null

    //test app package name
    private const val testPackageName = "com.embrace_2redbeans"

    private fun getVersionGooglePlay(): String? {
        try {
            val document: Document = Jsoup.connect(
                "https://play.google.com/store/apps/details?id=" + (if (!BuildConfig.DEBUG) BuildConfig.APPLICATION_ID else testPackageName) + "&hl=en"
            ).timeout(30000).userAgent("Mozilla/5.0 (Windows; U; WindowsNT 5.1; en-US; rv1.8.1.6) Gecko/20070725 Firefox/2.0.0.6").referrer("http://www.google.com").get()
            //            d(tag,"document:$document")
            val element: Elements = document.getElementsContainingOwnText("Current Version")
            for (ele in element) {
                Logg.d(tag, "ele:$ele")
                if (ele.siblingElements() != null) {
                    Logg.d(tag, "ele.siblingElements:${ele.siblingElements()}")
                    val sibElemets: Elements = ele.siblingElements()
                    for (sibElemet in sibElemets) {
                        newVersion = sibElemet.text()
                    }
                }
            }
        } catch (e: IOException) {
            e.printStackTrace()
        }
        return newVersion
    }


    /**
     * first->currentVersionName
     * second->mLatestVersionName
     * third->is not equal
     */
    fun startCheckVersion(): Triple<String?, String?, Boolean> {
        return try {
            val mLatestVersionName = getVersionGooglePlay()
            val currentVersionName = BuildConfig.VERSION_NAME
            Logg.d(tag, "versionName:$mLatestVersionName")
            Logg.d(tag, "VERSION_CODE:$currentVersionName")
            Triple(currentVersionName, mLatestVersionName, mLatestVersionName != currentVersionName)
        } catch (e: Exception) {
            e.printStackTrace()
            Triple(null, null, false)
        }
    }

    @SuppressLint("QueryPermissionsNeeded")
    fun jumpToGooglePlay(activity: Activity?) {
        activity?.let { it ->
            val intentData = Intent(Intent.ACTION_VIEW).setData(
                Uri.parse(
                    "market://details?id=" + if (!BuildConfig.DEBUG) it.packageName else testPackageName
                )
            )
            if (it.packageManager?.let { intentData.resolveActivity(it) } != null) {
                it.startActivity(intentData)
            } else {
                intentData.data = Uri.parse(
                    "https://play.google.com/store/apps/details?id=" + if (!BuildConfig.DEBUG) it.packageName else testPackageName
                )
                it.startActivity(intentData)
            }
        }
    }
}

调用

 coroutineScope?.launch(context = ioDispatcherDispatchers.IO) {
            try {
                onGetResponse(resultCallback, VersionUpdateUtil.startCheckVersion())
            } catch (e: Exception) {
                e.printStackTrace()
                onGetResponse(resultCallback, null)
            }
        }
        
 suspend fun <T> withNonCancellable(block: suspend CoroutineScope.() -> T): T {
        return withContext(NonCancellable, block)
    }

suspend fun <T> withMain(block: suspend CoroutineScope.() -> T): T {
        return withContext(mainDispatcher, block)
    }
 suspend fun <Data> onGetResponse(callback: (Data?) -> Unit, httpData: Data?) {
        callback.let {
            withNonCancellable {
                withMain {
                    it.invoke(httpData)
                }
            }
        }
    }

六、 尾声

获取HTML数据时,耗时操作要放入协程里面。目前的这个更新方式也有小缺点:google play哪天改了current version,那我们的app就没法获取最新版本更新了。

要在Android应用程序中实现Google反向地理编码,你可以使用Google Maps Geocoding API。下面是一个简单的示例代码,演示如何查找当前位置的地址信息: 首先,确保你的Android应用程序已经添加了Google Play服务库依赖。在项目的build.gradle文件中,添加以下依赖项: ```groovy implementation 'com.google.android.gms:play-services-maps:17.0.1' ``` 然后,在你的Activity或Fragment中,你可以使用以下代码来执行反向地理编码: ```java import android.Manifest; import android.content.pm.PackageManager; import android.location.Address; import android.location.Geocoder; import android.location.Location; import android.os.Bundle; import android.widget.Toast; import androidx.annotation.NonNull; import androidx.appcompat.app.AppCompatActivity; import androidx.core.app.ActivityCompat; import androidx.core.content.ContextCompat; import com.google.android.gms.location.FusedLocationProviderClient; import com.google.android.gms.location.LocationServices; import java.io.IOException; import java.util.List; import java.util.Locale; public class MainActivity extends AppCompatActivity { private static final int PERMISSIONS_REQUEST_ACCESS_FINE_LOCATION = 1; private FusedLocationProviderClient fusedLocationClient; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); fusedLocationClient = LocationServices.getFusedLocationProviderClient(this); if (ContextCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION) == PackageManager.PERMISSION_GRANTED) { getCurrentLocation(); } else { ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, PERMISSIONS_REQUEST_ACCESS_FINE_LOCATION); } } private void getCurrentLocation() { fusedLocationClient.getLastLocation() .addOnSuccessListener(this, new OnSuccessListener<Location>() { @Override public void onSuccess(Location location) { if (location != null) { Geocoder geocoder = new Geocoder(MainActivity.this, Locale.getDefault()); try { List<Address> addresses = geocoder.getFromLocation( location.getLatitude(), location.getLongitude(), 1 ); if (addresses.size() > 0) { Address address = addresses.get(0); String addressText = address.getAddressLine(0); Toast.makeText(MainActivity.this, addressText, Toast.LENGTH_SHORT).show(); } } catch (IOException e) { e.printStackTrace(); } } } }); } @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { if (requestCode == PERMISSIONS_REQUEST_ACCESS_FINE_LOCATION) { if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { getCurrentLocation(); } else { Toast.makeText(this, "Location permission denied", Toast.LENGTH_SHORT).show(); } } } } ``` 上述代码中,我们首先检查是否已经授予了定位权限。如果没有,我们请求用户授权。如果已经授予了权限,我们使用FusedLocationProviderClient获取当前位置的经纬度坐标。然后,我们使用Geocoder进行反向地理编码,获取当前位置的地址信息。最后,我们将地址信息显示在Toast中,你可以根据需要进行更改。 记得在AndroidManifest.xml文件中添加以下权限: ```xml <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" /> ``` 请注意,这只是一个简单的示例,你可能需要根据自己的应用程序需求进行适当的修改和错误处理。另外,确保在使用Google Maps Geocoding API时遵守相关的服务条款和政策。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值